👀 Reading hidden code
using CellularAutomata
👀 Reading hidden code
using PlutoUI
PlutoUI.ExperimentalLayout
👀 Reading hidden code
👀 Reading hidden code
using HypertextLiteral
: @htl, @htl_str
Demo
👀 Reading hidden code
ncells:
👀 Reading hidden code
generations:
👀 Reading hidden code
rule:
👀 Reading hidden code
30
👀 Reading hidden code
👀 Reading hidden code
show_border:
👀 Reading hidden code
👀 Reading hidden code
false
false
false
false
false
false
false
false
false
false
false
false
false
false
false
false
false
false
false
false
false
false
false
false
false
false
false
false
false
false
starting_val = generate_starting_val(Bool, ncells)
👀 Reading hidden code
30
30
0
1
1
1
1
0
2
1
30×111 Matrix{Bool}: 0 0 0 0 0 0 0 0 0 0 0 0 0 0 … 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 … 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 ⋮ ⋮ ⋮ ⋱ ⋮ ⋮ ⋮ 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 … 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
ca = CellularAutomaton(DCA(rule), starting_val, generations)
👀 Reading hidden code
👀 Reading hidden code
Utility functions
👀 Reading hidden code
generate_starting_val (generic function with 1 method)
👀 Reading hidden code
demo_ca (generic function with 3 methods)
👀 Reading hidden code
👀 Reading hidden code
Display automata sequence
👀 Reading hidden code
demo_ca(DCA(30)) |> show_sequence
👀 Reading hidden code
demo_ca(DCA(7110222193934; states=3, radius=1), 121, 50) |> show_sequence
👀 Reading hidden code
show_sequence (generic function with 1 method)
show_sequence(ca; border::Bool=false) = @htl """
<div
class="plui-cell-automata $(border ? "border" : "")"
style=$(Dict(
:display => "grid",
:grid_template_columns => "repeat($(size(ca.evolution, 2)), auto)",
:contain => "layout paint",
))>$(
Iterators.map(PermutedDimsArray(ca.evolution, (2,1))) do x
@htl "<div style=$(Dict(
:background => background(ca, x),
))></div>"
end
)</div>
<style>
.plui-cell-automata > div {
aspect-ratio: 1;
}
.plui-cell-automata.border > div {
border: .2px solid #888;
}
</style>
"""
👀 Reading hidden code
background (generic function with 1 method)
background(ca::CellularAutomaton{<:CCA}, x::Float64) = "hsl(0deg,0%,$(x*100)%)"
👀 Reading hidden code
background (generic function with 2 methods)
background(ca::CellularAutomaton{<:DCA}, x::Integer) =
"hsl(0deg,0%,$(100.0 * (1.0 - Float64(x) / (ca.generation_fun.states - 1)))%)"
👀 Reading hidden code
background (generic function with 3 methods)
background(ca::CellularAutomaton{<:DCA}, x::Bool) = x ? "black" : "white"
👀 Reading hidden code
show_sequence_old(ca ) = Layout.grid(
map(ca.evolution) do x
@htl "<div style=$(Dict(
"background" => x ? "black" : "white",
"border" => ".2px solid #888",
# "width" => "1em",
# "height" => "1em",
"aspect-ratio" => "1",
))></div>"
end;
column_gap="0",
style=Dict("background" => "white", "contain" => "layout paint")
)
👀 Reading hidden code
👀 Reading hidden code
Interactive rule input
👀 Reading hidden code
Single T input
Simple idea: the interactive bottom bit is a PlutoUI.CheckBox
. We can use PlutoUI.Experimental.wrapped
to display a the checkbox with other elements around it.
👀 Reading hidden code
T_input (generic function with 2 methods)
T_input(a, b, c, default=false) = PlutoUI.Experimental.wrapped() do Child
Layout.grid([
disabled_checkbox(a) disabled_checkbox(b) disabled_checkbox(c)
Text("") Child(CheckBox(default)) Text("")
]; fill_width=false, column_gap="0")
end
👀 Reading hidden code
@bind z T_input(true, false, true)
👀 Reading hidden code
false
z
👀 Reading hidden code
👀 Reading hidden code
disabled_checkbox (generic function with 1 method)
👀 Reading hidden code
👀 Reading hidden code
Combining 8 T inputs
PlutoUI.combine
lets us combine multiple inputs into one that you can bind to.
👀 Reading hidden code
@bind zzz PlutoUI.combine() do Child
Ts = Iterators.map(1:8) do i
# a,b,c are the top 3 values of the T
# These are the binary representations of the numbers [0, ..., 7]
abc = (c == '1' for c in bitstring(8-i)[end-2:end])
Child(T_input(abc...))
end
# display the Ts horizontally
Layout.hbox(Ts; style=Dict(:gap => "1em"))
end
👀 Reading hidden code
false
false
false
false
false
false
false
false
zzz
👀 Reading hidden code
👀 Reading hidden code
Converting between bits and the rule number
👀 Reading hidden code
ruletobits (generic function with 1 method)
👀 Reading hidden code
false
false
false
true
true
true
true
false
collect(ruletobits(30))
👀 Reading hidden code
👀 Reading hidden code
bitstorule (generic function with 1 method)
👀 Reading hidden code
30
bitstorule(ruletobits(30))
👀 Reading hidden code
👀 Reading hidden code
Putting it all together
We add the two transformations:
For the initial value: rule number to bits. These bits are the initial values of the checkboxes
For the
@bind
value: bits to rule number. We usePlutoUI.Experimental.transformed_value
to apply a transformation function to the value returned via@bind
.
👀 Reading hidden code
rule_input (generic function with 1 method)
function rule_input(; default::Integer)
# default values of the interactive checkboxes, i.e. the bottoms of the Ts
default_checked = ruletobits(default)
# the 8 interactive Ts are combined into a single bindable element
combined = PlutoUI.combine() do Child
Ts = Iterators.map(enumerate(default_checked)) do (i, default)
# a,b,c are the top 3 values of the T
# These are the binary representations of the numbers [0, ..., 7]
abc = (c == '1' for c in bitstring(8-i)[end-2:end])
Child(T_input(abc..., default))
end
# display the Ts horizontally
Layout.hbox(Ts; style=Dict(:gap => "1em"))
end
# convert the selected bits into the rule number:
PlutoUI.Experimental.transformed_value(bitstorule, combined)
end
👀 Reading hidden code
@bind myrule rule_input(default=30)
👀 Reading hidden code
30
myrule
👀 Reading hidden code
👀 Reading hidden code
👀 Reading hidden code
Macro @bindname
👀 Reading hidden code
zzze:
@bindname zzze Slider(1:100)
👀 Reading hidden code
1
zzze
👀 Reading hidden code
@bindname
Like @bind
in Pluto, but it also displays the name of the variable.
"""
Like `@bind` in Pluto, but it also displays the name of the variable.
"""
macro bindname(name::Symbol, ex::Expr)
quote
HypertextLiteral.@htl("""
<div style='display: flex; flex-wrap: wrap; align-items: baseline;'>
<code style='font-weight: bold'>$($(String(name))):</code> $(@bind $(name) $(esc(ex)))
</div>
""")
end
end
👀 Reading hidden code
rulasdflkjasdlkfjasldkjfklajsdflkjasdfeee:
@bindname rulasdflkjasdlkfjasldkjfklajsdflkjasdfeee rule_input(default=30)
👀 Reading hidden code
kjdfjkh:
@bindname kjdfjkh rule_input(default=30)
👀 Reading hidden code